home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 1 / CU Amiga Magazine CD-ROM Special Edition (1995)(EMAP Images)(GB)[Issue 1995-11].iso / Aminet / text / edit / WarpMail11.lha / WarpMail / source / lib.c < prev    next >
C/C++ Source or Header  |  1995-04-09  |  4KB  |  164 lines

  1. /*
  2.  *  LIB.C
  3.  *
  4.  *  Basic Library Resource Handling
  5.  *
  6.  *  NOTE: all data declarations should be initialized since we skip
  7.  *        normal C startup code (unless initial value is don't care)
  8.  *
  9.  *  WARNING: arguments are passed in certain registers from the assembly
  10.  *        tag file, matched to how they are declared below.  Do not change
  11.  *        the argument declarations!
  12.  */
  13.  
  14. #include "defs.h"
  15.  
  16. Prototype LibCall struct Library *LibInit   (__D0 BPTR);
  17. Prototype LibCall struct Library *LibOpen   (__D0 long, __A0 struct Library *);
  18. Prototype LibCall long            LibClose  (__A0 struct Library *);
  19. Prototype LibCall long            LibExpunge(__A0 struct Library *);
  20.  
  21. struct Library *LibBase = NULL;
  22.  
  23. long SysBase  = NULL;
  24. long DOSBase  = NULL;
  25. BPTR SegList  = 0;
  26.  
  27. /*
  28.  *    The Initialization routine is given only a seglist pointer.  Since
  29.  *    we are NOT AUTOINIT we must construct and add the library ourselves
  30.  *    and return either NULL or the library pointer.  Exec has Forbid()
  31.  *    for us during the call.
  32.  *
  33.  *    We use an extended library structure to allow identification as a
  34.  *    GoldED syntax scanner.
  35.  */
  36.  
  37. LibCall struct Library *
  38. LibInit(__D0 BPTR segment)
  39. {
  40.     struct Library *lib;
  41.  
  42.     static const long Vectors[] = {
  43.  
  44.         (long)ALibOpen,
  45.         (long)ALibClose,
  46.         (long)ALibExpunge,
  47.         (long)ALibReserved,
  48.  
  49.         (long)MountScanner,
  50.         (long)StartScanner,
  51.         (long)CloseScanner,
  52.         (long)FlushScanner,
  53.         (long)SetupScanner,
  54.         (long)BriefScanner,
  55.         (long)ParseLine,
  56.         (long)UnparseLines,
  57.         (long)ParseSection,
  58.         -1
  59.     };
  60.  
  61.     SysBase = *(long *)4;
  62.  
  63.     if (DOSBase = OpenLibrary("dos.library", 0)) {
  64.  
  65.         if (LibBase = lib = MakeLibrary((APTR)Vectors, NULL, NULL, sizeof(struct ParserBase), NULL)) {
  66.  
  67.             lib->lib_Node.ln_Type = NT_LIBRARY;
  68.             lib->lib_Node.ln_Name = LibName;
  69.             lib->lib_Flags        = LIBF_CHANGED | LIBF_SUMUSED;
  70.             lib->lib_Version      = 37;
  71.             lib->lib_Revision     = 0;
  72.             lib->lib_IdString     = (APTR)LibId;
  73.  
  74.             ((struct ParserBase *)lib)->Magic = PARSER_MAGIC;
  75.  
  76.             SegList = segment;
  77.  
  78.             AddLibrary(lib);
  79.  
  80.             InitC();
  81.  
  82.             return(lib);
  83.         }
  84.     }
  85.  
  86.     return(NULL);
  87. }
  88.  
  89. /*
  90.  *    Open is given the library pointer and the version request.  Either
  91.  *    return the library pointer or NULL.  Remove the DELAYED-EXPUNGE flag.
  92.  *    Exec has Forbid() for us during the call.
  93.  */
  94.  
  95. LibCall struct Library *
  96. LibOpen(__D0 long version, __A0 struct Library *lib)
  97. {
  98.     ++lib->lib_OpenCnt;
  99.  
  100.     lib->lib_Flags &= ~LIBF_DELEXP;
  101.  
  102.     return(lib);
  103. }
  104.  
  105. /*
  106.  *    Close is given the library pointer and the version request.  Be sure
  107.  *    not to decrement the open count if already zero.  If the open count
  108.  *    is or becomes zero AND there is a LIBF_DELEXP, we expunge the library
  109.  *    and return the seglist.  Otherwise we return NULL.
  110.  *
  111.  *    Note that this routine never sets LIBF_DELEXP on its own.
  112.  *
  113.  *    Exec has Forbid() for us during the call.
  114.  */
  115.  
  116. LibCall long
  117. LibClose(__A0 struct Library *lib)
  118. {
  119.     if (lib->lib_OpenCnt && --lib->lib_OpenCnt)
  120.         return(NULL);
  121.  
  122.     if (lib->lib_Flags & LIBF_DELEXP)
  123.         return(LibExpunge(lib));
  124.  
  125.     return(NULL);
  126. }
  127.  
  128. /*
  129.  *    We expunge the library and return the Seglist ONLY if the open count
  130.  *    is zero.  If the open count is not zero we set the DELAYED-EXPUNGE
  131.  *    flag and return NULL.
  132.  *
  133.  *    Exec has Forbid() for us during the call.  NOTE ALSO that Expunge
  134.  *    might be called from the memory allocator and thus we CANNOT DO A
  135.  *    Wait() or otherwise take a long time to complete (straight from RKM).
  136.  *
  137.  *    Apparently RemLibrary(lib) calls our expunge routine and would
  138.  *    therefore freeze if we called it ourselves.  As far as I can tell
  139.  *    from RKM, LibExpunge(lib) must remove the library itself as shown
  140.  *    below.
  141.  */
  142.  
  143. LibCall long
  144. LibExpunge(__A0 struct Library *lib)
  145. {
  146.     if (lib->lib_OpenCnt) {
  147.  
  148.         lib->lib_Flags |= LIBF_DELEXP;
  149.         return(NULL);
  150.     }
  151.  
  152.     Remove(&lib->lib_Node);
  153.  
  154.     FreeMem((char *)lib - lib->lib_NegSize, lib->lib_NegSize + lib->lib_PosSize);
  155.  
  156.     if (DOSBase) {
  157.  
  158.         CloseLibrary((struct Library *)DOSBase);
  159.         DOSBase = NULL;
  160.     }
  161.  
  162.     return((long)SegList);
  163. }
  164.